#pragma once
#include <iostream>
#include <pthread.h>
#include <queue>
#include <unistd.h>
#include "Task.hpp"

#define NUM 5

template <typename T>
class BlockQueue
{
private:
    bool IsEmpty()
    {
        return _q.empty();
    }
    bool IsFull()
    {
        return _q.size() == _cap;
    }

public:
    BlockQueue(int cap = NUM)
        : _cap(cap)
    {
        pthread_mutex_init(&_mutex, nullptr);
        pthread_cond_init(&_full, nullptr);
        pthread_cond_init(&_empty, nullptr);
    }
    ~BlockQueue()
    {
        pthread_mutex_destroy(&_mutex);
        pthread_cond_destroy(&_full);
        pthread_cond_destroy(&_empty);
    }
    // 向阻塞队列插入数据(生产者调用)
    void Push(const T& data)
    {
        pthread_mutex_lock(&_mutex);
        while (IsFull())
        {
            // 不能进行生产，直到阻塞队列可以容纳新的数据
            pthread_cond_wait(&_full, &_mutex);
        }
        _q.push(data);
        if (_q.size() >= _cap / 2)
            pthread_cond_signal(&_empty); // 唤醒在条件变量empty下等待的消费者线程
        pthread_mutex_unlock(&_mutex);
    }
    // 从阻塞队列获取数据(消费者调用)
    void Pop(T* data)
    {
        pthread_mutex_lock(&_mutex);
        while (IsEmpty())
        {
            // 不能进行消费，直到阻塞队列当中有数据
            pthread_cond_wait(&_empty, &_mutex);
        }
        *data = _q.front();
        _q.pop();
        if (_q.size() <= _cap / 2)
            pthread_cond_signal(&_full); // 唤醒在条件变量full下等待的生产者线程
        pthread_mutex_unlock(&_mutex);
    }

private:
    std::queue<T> _q; // 阻塞队列
    int _cap;         // 阻塞队列最大容量
    pthread_mutex_t _mutex;
    pthread_cond_t _full;
    pthread_cond_t _empty;
};